/*
 *  Copyright (c) 2008 Stephan Arts <stephan@xfce.org>
 *  Copyright (c) 2008 Brian Tarricone <bjt23@cornell.edu>
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; version 2 of the License ONLY.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif

#ifdef HAVE_SYS_STAT_H
#include <sys/stat.h>
#endif

#ifdef HAVE_ERRNO_H
#include <errno.h>
#endif

#ifdef HAVE_FCNTL_H
#include <fcntl.h>
#endif

#ifdef HAVE_STDLIB_H
#include <stdlib.h>
#endif

#ifdef HAVE_STRING_H
#include <string.h>
#endif

#include <glib.h>
#include <libxfce4util/libxfce4util.h>
#include <stdio.h>

#include "common/xfconf-common-private.h"
#include "common/xfconf-gvaluefuncs.h"
#include "xfconf/xfconf.h"

static gboolean version = FALSE;
static gboolean list = FALSE;
static gboolean verbose = FALSE;
static gboolean create = FALSE;
static gboolean reset = FALSE;
static gboolean recursive = FALSE;
static gboolean force_array = FALSE;
static gboolean monitor = FALSE;
static gboolean toggle = FALSE;
static gchar *channel_name = NULL;
static gchar *property_name = NULL;
static gchar **set_value = NULL;
static gchar **type = NULL;

static void
xfconf_query_monitor(XfconfChannel *channel, const gchar *changed_property, GValue *property_value)
{
    gchar *string;

    if (property_name && !g_str_has_prefix(changed_property, property_name)) {
        return;
    }

    if (property_value && G_IS_VALUE(property_value)) {
        if (verbose) {
            string = _xfconf_string_from_gvalue(property_value);
            g_print("%s: %s (%s)\n", _("set"), changed_property, string);
            g_free(string);
        } else {
            g_print("%s: %s\n", _("set"), changed_property);
        }
    } else {
        g_print("%s: %s\n", _("reset"), changed_property);
    }
}

static void
xfconf_query_get_propname_size(gpointer key, gpointer value, gpointer user_data)
{
    gint *size = user_data;
    gchar *propname = (gchar *)key;

    if ((gint)strlen(propname) > *size) {
        *size = strlen(propname);
    }
}

static void
xfconf_query_list_sorted(gpointer key, gpointer value, gpointer user_data)
{
    GSList **listp = user_data;

    *listp = g_slist_insert_sorted(*listp, key, (GCompareFunc)g_utf8_collate);
}

static void
xfconf_query_list_contents(GSList *sorted_contents, GHashTable *channel_contents, gint size)
{
    GSList *li;
    gchar *format = verbose ? g_strdup_printf("%%-%ds%%s\n", size + 2) : NULL;
    GValue *property_value;
    gchar *string;

    for (li = sorted_contents; li != NULL; li = li->next) {
        if (verbose) {
            property_value = g_hash_table_lookup(channel_contents, li->data);

            if (G_TYPE_PTR_ARRAY != G_VALUE_TYPE(property_value)) {
                string = _xfconf_string_from_gvalue(property_value);
            } else {
                GPtrArray *arr = g_value_get_boxed(property_value);
                gchar **strv = g_new0(gchar *, arr->len + 1);
                gchar *str;

                for (guint i = 0; i < arr->len; ++i) {
                    GValue *item_value = g_ptr_array_index(arr, i);
                    strv[i] = _xfconf_string_from_gvalue(item_value);
                }
                str = g_strjoinv(",", strv);
                string = g_strdup_printf("[%s]", str);
                g_free(str);
                g_strfreev(strv);
            }

            g_print(format, (gchar *)li->data, string);
            g_free(string);
        } else {
            g_print("%s\n", (gchar *)li->data);
        }
    }

    g_free(format);
}

static void
xfconf_query_printerr(const gchar *message,
                      ...)
{
    va_list args;
    gchar *str;

    va_start(args, message);
    str = g_strdup_vprintf(message, args);
    va_end(args);

    g_printerr("%s.\n", str);
    g_free(str);
}

static gint
xfconf_query_compare_func(gconstpointer a,
                          gconstpointer b,
                          gpointer user_data)
{
    gchar **s = (gchar **)a;
    gchar **t = (gchar **)b;
    return g_strcmp0(*s, *t);
}

static GOptionEntry entries[] = {
    { "version", 'V', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &version,
      N_("Version information"),
      NULL },
    { "channel", 'c', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING, &channel_name,
      N_("The channel to query/modify"),
      NULL },
    { "property", 'p', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING, &property_name,
      N_("The property to query/modify"),
      NULL },
    { "set", 's', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING_ARRAY, &set_value,
      N_("The new value to set for the property"),
      NULL },
    { "list", 'l', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &list,
      N_("List properties (or channels if -c is not specified)"),
      NULL },
    { "verbose", 'v', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &verbose,
      N_("Print property and value in combination with -l or -m"),
      NULL },
    { "create", 'n', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &create,
      N_("Create a new property if it does not already exist"),
      NULL },
    { "type", 't', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_STRING_ARRAY, &type,
      N_("Specify the property value type"),
      NULL },
    { "reset", 'r', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &reset,
      N_("Reset property"),
      NULL },
    { "recursive", 'R', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &recursive,
      N_("Recursive (use with -r)"),
      NULL },
    { "force-array", 'a', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &force_array,
      N_("Force array even if only one element"),
      NULL },
    { "toggle", 'T', G_OPTION_FLAG_IN_MAIN, G_OPTION_ARG_NONE, &toggle,
      N_("Invert an existing boolean property"),
      NULL },
    {
        "monitor",
        'm',
        G_OPTION_FLAG_IN_MAIN,
        G_OPTION_ARG_NONE,
        &monitor,
        N_("Monitor a channel for property changes"),
        NULL,
    },
    { NULL }
};

int
main(int argc, char **argv)
{
    GError *error = NULL;
    XfconfChannel *channel = NULL;
    gboolean prop_exists;
    GOptionContext *context;

    xfce_textdomain(GETTEXT_PACKAGE, LOCALEDIR, "UTF-8");

    context = g_option_context_new(_("- Xfconf commandline utility"));
    g_option_context_add_main_entries(context, entries, GETTEXT_PACKAGE);

    if (!g_option_context_parse(context, &argc, &argv, &error)) {
        xfconf_query_printerr(_("Option parsing failed: %s"), error->message);
        g_error_free(error);
        g_option_context_free(context);
        return EXIT_FAILURE;
    }
    g_option_context_free(context);

    if (version) {
        g_print("xfconf-query");
        g_print(" %s\n\n", PACKAGE_VERSION);
        g_print("%s\n", "Copyright (c) 2008-2024");
        g_print("\t%s\n\n", _("The Xfce development team. All rights reserved."));
        g_print(_("Please report bugs to <%s>."), PACKAGE_BUGREPORT);
        g_print("\n");

        return EXIT_SUCCESS;
    }

    /** Check if the property is specified */
    if (!property_name && !list && !monitor) {
        xfconf_query_printerr(_("No property specified"));
        return EXIT_FAILURE;
    }

    if (create && reset) {
        xfconf_query_printerr(_("--create and --reset options can not be used together"));
        return EXIT_FAILURE;
    }

    if ((create || reset) && (list)) {
        xfconf_query_printerr(_("--create and --reset options can not be used together with --list"));
        return EXIT_FAILURE;
    }

    if (!xfconf_init(&error)) {
        xfconf_query_printerr(_("Failed to init libxfconf: %s"), error->message);
        g_error_free(error);
        return EXIT_FAILURE;
    }

    /** Check if the channel is specified */
    if (!channel_name) {
        gchar **channels;
        gint i;

        g_print("%s\n", _("Channels:"));

        channels = xfconf_list_channels();
        if (G_LIKELY(channels)) {
            g_qsort_with_data(channels, g_strv_length(channels), sizeof(gchar *), xfconf_query_compare_func, NULL);
            for (i = 0; channels[i]; ++i) {
                g_print("  %s\n", channels[i]);
            }
            g_strfreev(channels);
        } else {
            xfconf_shutdown();
            return EXIT_FAILURE;
        }
        xfconf_shutdown();
        return EXIT_SUCCESS;
    }

    channel = xfconf_channel_new(channel_name);

    if (monitor) {
        GMainLoop *loop;

        g_signal_connect(G_OBJECT(channel), "property-changed", G_CALLBACK(xfconf_query_monitor), NULL);

        g_print(_("Start monitoring channel \"%s\":"), channel_name);
        g_print("\n\n");

        loop = g_main_loop_new(NULL, TRUE);
        g_main_loop_run(loop);
        g_main_loop_unref(loop);
    }

    if (property_name && !list) {
        if (reset) {
            /** Reset property */
            xfconf_channel_reset_property(channel, property_name, recursive);
        } else if (set_value == NULL || set_value[0] == NULL) {
            /** Read value */
            GValue value = G_VALUE_INIT;

            if (!xfconf_channel_get_property(channel, property_name, &value)) {
                xfconf_query_printerr(_("Property \"%s\" does not exist on channel \"%s\""),
                                      property_name, channel_name);
                xfconf_shutdown();
                return EXIT_FAILURE;
            }

            if (toggle) {
                if (G_VALUE_HOLDS_BOOLEAN(&value)) {
                    if (xfconf_channel_set_bool(channel, property_name, !g_value_get_boolean(&value))) {
                        xfconf_shutdown();
                        return EXIT_SUCCESS;
                    } else {
                        xfconf_query_printerr(_("Failed to set property"));
                    }
                } else {
                    xfconf_query_printerr(_("--toggle only works with boolean values"));
                }
                xfconf_shutdown();
                return EXIT_FAILURE;
            }

            if (G_TYPE_PTR_ARRAY != G_VALUE_TYPE(&value)) {
                gchar *str_val = _xfconf_string_from_gvalue(&value);
                g_print("%s\n", str_val ? str_val : _("(unknown)"));
                g_free(str_val);
                g_value_unset(&value);
            } else {
                GPtrArray *arr = g_value_get_boxed(&value);
                guint i;

                g_print(_("Value is an array with %d items:"), arr->len);
                g_print("\n\n");

                for (i = 0; i < arr->len; ++i) {
                    GValue *item_value = g_ptr_array_index(arr, i);

                    if (item_value) {
                        gchar *str_val = _xfconf_string_from_gvalue(item_value);
                        g_print("%s\n", str_val ? str_val : _("(unknown)"));
                        g_free(str_val);
                    }
                }
            }
        } else {
            /* Write value */
            GValue value = G_VALUE_INIT;
            gint i;

            prop_exists = xfconf_channel_has_property(channel, property_name);
            if (!prop_exists && !create) {
                xfconf_query_printerr(_("Property \"%s\" does not exist on channel \"%s\". If a new "
                                        "property should be created, use the --create option"),
                                      property_name, channel_name);
                xfconf_shutdown();
                return EXIT_FAILURE;
            }

            if (!prop_exists && (!type || !type[0])) {
                xfconf_query_printerr(_("When creating a new property, the value type must be specified"));
                xfconf_shutdown();
                return EXIT_FAILURE;
            }

            if (prop_exists && !(type && type[0])) {
                /* only care what the existing type is if the user
                 * didn't specify one */
                if (!xfconf_channel_get_property(channel, property_name, &value)) {
                    xfconf_query_printerr(_("Failed to get the existing type for the value"));
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }
            }

            if (!set_value[1] && !force_array) {
                /* not an array */
                GType gtype = G_TYPE_INVALID;

                if (prop_exists) {
                    gtype = G_VALUE_TYPE(&value);
                }

                if (type && type[0]) {
                    gtype = _xfconf_gtype_from_string(type[0]);
                }

                if (G_TYPE_INVALID == gtype || G_TYPE_NONE == gtype) {
                    xfconf_query_printerr(_("Unable to determine the type of the value"));
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }

                if (G_TYPE_PTR_ARRAY == gtype) {
                    xfconf_query_printerr(_("A value type must be specified to change an array into a single value"));
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }

                if (G_VALUE_TYPE(&value)) {
                    g_value_unset(&value);
                }

                g_value_init(&value, gtype);
                if (!_xfconf_gvalue_from_string(&value, set_value[0])) {
                    xfconf_query_printerr(_("Unable to convert \"%s\" to type \"%s\""),
                                          set_value[0], g_type_name(gtype));
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }

                if (!xfconf_channel_set_property(channel, property_name, &value)) {
                    xfconf_query_printerr(_("Failed to set property"));
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }
            } else {
                /* array property */
                GPtrArray *arr_old = NULL, *arr_new;
                gint new_values = 0, new_types = 0;

                for (new_values = 0; set_value[new_values]; ++new_values)
                    ;
                if (type && type[0]) {
                    for (new_types = 0; type[new_types]; ++new_types)
                        ;
                } else if (G_VALUE_TYPE(&value) == G_TYPE_PTR_ARRAY) {
                    arr_old = g_value_get_boxed(&value);
                    new_types = arr_old->len;
                }

                if (new_values != new_types) {
                    xfconf_query_printerr(_("There are %d new values, but only %d types could be determined"),
                                          new_values, new_types);
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }

                arr_new = g_ptr_array_sized_new(new_values);

                for (i = 0; set_value[i]; ++i) {
                    GType gtype = G_TYPE_INVALID;
                    GValue *value_new;

                    if (arr_old) {
                        GValue *value_old = g_ptr_array_index(arr_old, i);
                        gtype = G_VALUE_TYPE(value_old);
                    } else {
                        gtype = _xfconf_gtype_from_string(type[i]);
                    }

                    if (G_TYPE_INVALID == gtype || G_TYPE_NONE == gtype) {
                        xfconf_query_printerr(_("Unable to determine type of value at index %d"), i);
                        xfconf_shutdown();
                        return EXIT_FAILURE;
                    }

                    value_new = g_new0(GValue, 1);
                    g_value_init(value_new, gtype);
                    if (!_xfconf_gvalue_from_string(value_new, set_value[i])) {
                        xfconf_query_printerr(_("Unable to convert \"%s\" to type \"%s\""),
                                              set_value[i], g_type_name(gtype));
                        xfconf_shutdown();
                        return EXIT_FAILURE;
                    }
                    g_ptr_array_add(arr_new, value_new);
                }

                if (G_VALUE_TYPE(&value)) {
                    g_value_unset(&value);
                }

                g_value_init(&value, G_TYPE_PTR_ARRAY);
                g_value_set_boxed(&value, arr_new);

                if (!xfconf_channel_set_property(channel, property_name, &value)) {
                    xfconf_query_printerr(_("Failed to set property"));
                    xfconf_shutdown();
                    return EXIT_FAILURE;
                }
            }
        }
    }

    if (list) {
        GHashTable *channel_contents = xfconf_channel_get_properties(channel, property_name);
        if (channel_contents) {
            gint size = 0;
            GSList *sorted_contents = NULL;

            if (verbose) {
                g_hash_table_foreach(channel_contents, (GHFunc)xfconf_query_get_propname_size, &size);
            }

            g_hash_table_foreach(channel_contents, (GHFunc)xfconf_query_list_sorted, &sorted_contents);

            xfconf_query_list_contents(sorted_contents, channel_contents, size);

            g_slist_free(sorted_contents);
            g_hash_table_destroy(channel_contents);
        } else {
            g_print(_("Channel \"%s\" contains no properties"), channel_name);
            g_print(".\n");
        }
    }

    xfconf_shutdown();

    return EXIT_SUCCESS;
}
